Completed
Push — master ( 65279b...663c4c )
by Andres
29s
created

angular.controller(ꞌct_redoxꞌ)   A

Complexity

Conditions 2
Paths 2

Size

Total Lines 14

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
cc 2
c 0
b 0
f 0
nc 2
dl 0
loc 14
rs 9.4285
nop 2
1
/**
2
 redox
3
 Component that handles reduction/oxidation and ions.
4
5
 @namespace Components
6
 */
7
'use strict';
8
9
angular.module('game').component('redox', {
10
  templateUrl: 'views/redox.html',
11
  controller: 'ct_redox',
12
  controllerAs: 'ct'
13
});
14
15
angular.module('game').controller('ct_redox', ['state', 'data', 'visibility', 'util', 'format', 'reaction', 'upgrade',
16
  function (state, data, visibility, util, format, reaction, upgradeService) {
17
    let ct = this;
18
    ct.state = state;
19
    ct.data = data;
20
    ct.util = util;
21
    ct.format = format;
22
    ct.reaction = reaction;
23
    ct.upgradeService = upgradeService;
24
    ct.adjustAmount = [1, 10, 25, 100];
25
26
    ct.update = function(player) {
27
      processRedox(player);
28
      processElectronegativity(player);
29
    };
30
31
    function processRedox(player){
32
      for(let slot of player.element_slots){
33
        if(!slot){
34
          continue;
35
        }
36
        for (let redox of slot.redoxes) {
37
          if (!redox.resource || !redox.active || redox.from === redox.to) {
38
            continue;
39
          }
40
41
          let reactant = ct.generateName(redox.element, redox.from);
42
          let power = util.calculateValue(data.global_upgrades.redox_bandwidth.power.base,
43
                data.global_upgrades.redox_bandwidth.power,
44
                player.global_upgrades_current.redox_bandwidth);
45
          let number = Math.min(power, player.resources[reactant].number);
46
          let react = ct.redoxReaction(redox);
47
48
          ct.reaction.react(number, react, player);
49
        }
50
      }
51
    }
52
53
    function processElectronegativity(player){
54
      for(let key in data.elements){
55
        let element = data.elements[key];
56
        if(element.electronegativity === 0){
57
          continue;
58
        }
59
        let ions = element.anions.concat(element.cations);
60
		    ions.push(element.main);
61
        for(let resource of ions){
62
          if(player.resources[resource].number === 0){
63
            continue;
64
          }
65
          let charge = data.resources[resource].charge || 0;
66
          let probabilities = probabilityDistribution(key, charge);
67
          for(let probKey in probabilities){
68
            if(probKey === charge){
69
              continue;
70
            }
71
            let production = Math.floor(probabilities[probKey]*player.resources[resource].number);
72
            if(production === 0){
73
              continue;
74
            }
75
            let react = ct.redoxReaction({
76
              element: key,
77
              from: charge,
78
              to: parseInt(probKey, 10)
79
            });
80
            // electronegativity is 'for free'
81
      			react.reactant.eV = 0;
82
            // FIXME: starvation should fix this
83
            if(react.reactant['e-']){
84
              production = Math.min(production, player.resources['e-'].number);
85
            }
86
            ct.reaction.react(production, react, player);
87
          }
88
        }
89
      }
90
    }
91
92
    function probabilityDistribution(element, charge){
93
    	let prob = {};
94
    	let start = -data.elements[element].electron_affinity.length;
95
    	let end = charge;
96
    	// lower than index, affected by negativity
97
    	rangeProbability(element, prob, start, end, 1, data.elements[element].negative_factor);
98
99
    	prob[charge] = 1;
100
101
    	start = charge+1;
102
    	end = data.elements[element].ionization_energy.length+1;
103
    	// lower than index, affected by positivity
104
    	rangeProbability(element, prob, start, end, -1, data.elements[element].positive_factor);
105
106
    	let sum = 0;
107
    	for(let i in prob){
108
    		sum += prob[i];
109
    	}
110
    	for(let i in prob){
111
    		prob[i] /= sum;
112
    	}
113
    	return prob;
114
    }
115
116
    function rangeProbability(element, prob, start, end, offset, factor){
117
    	for(let i = start; i < end; i++){
118
    		let difference = data.redox[element][i]-data.redox[element][i+offset];
119
    		if(difference <= 0){
120
    			difference = -difference;
121
    		}else{
122
    			difference = 1/difference;
123
    		}
124
    		prob[i] = Math.pow(data.constants.ELECTRONEGATIVITY_CHANCE,Math.abs(i))*factor*difference;
125
    	}
126
    }
127
128
    /* Writes a redox in the form of a reaction so that we can use the reaction
129
    service to process it */
130
    ct.redoxReaction = function (redox) {
131
      let reactant = ct.generateName(redox.element, redox.from);
132
      let product = ct.generateName(redox.element, redox.to);
133
      let energy = redoxEnergy(redox.from, redox.to, redox.element);
134
135
      let react = {
136
        'reactant': {},
137
        'product': {}
138
      };
139
140
      react.reactant[reactant] = 1;
141
      react.product[product] = 1;
142
      if (energy > 0) {
143
        react.reactant.eV = energy;
144
      } else if (energy < 0) {
145
        react.product.eV = -energy;
146
      }
147
148
      let electron = redox.from - redox.to;
149
      if (electron > 0) {
150
        react.reactant['e-'] = electron;
151
      } else if (electron < 0) {
152
        react.product['e-'] = -electron;
153
      }
154
155
      return react;
156
    };
157
158
    /* Calculates how much energy it takes to go from a redox level to another
159
    for a given element */
160
    function redoxEnergy(from, to, element) {
161
      let energyFrom = data.redox[element][from];
162
      let energyTo = data.redox[element][to];
163
      let energy = energyTo - energyFrom;
164
165
      return energy;
166
    }
167
168
    /* Generates the name of a ion, e.g. O3+ */
169
    ct.generateName = function (element, i) {
170
      if (i === 0) {
171
        return data.elements[element].main;
172
      }
173
      let postfix = '';
174
      if (Math.abs(i) > 1) {
175
        postfix = Math.abs(i);
176
      }
177
      postfix += getSign(i);
178
      let name = element + postfix;
179
      return name;
180
    };
181
182
    function getSign(number) {
183
      return number > 0 ? '+' : '-';
184
    }
185
186
    ct.redoxSize = function (player) {
187
      let size = 0;
188
      for(let slot of player.element_slots){
189
        if(!slot){
190
          continue;
191
        }
192
        size += slot.redoxes.length;
193
      }
194
      return size;
195
    };
196
197
    /* Adds a new redox to the player list */
198
    ct.addRedox = function (player, slot) {
199
      if(ct.redoxSize(player) >= util.calculateValue(data.global_upgrades.redox_slots.power.base,
200
            data.global_upgrades.redox_slots.power,
201
            player.global_upgrades.redox_slots)){
202
        return;
203
      }
204
      slot.redoxes.push({
205
        resource: data.elements[slot.element].main,
206
        active: false,
207
        element: slot.element,
208
        from: 0,
209
        to: 1
210
      });
211
    };
212
213
    ct.removeRedox = function (slot, index) {
214
      slot.redoxes.splice(index, 1);
215
    };
216
217
    ct.visibleRedox = function(slot) {
218
      return slot.redoxes;
219
    };
220
221
    ct.adjustLevel = function(player, upgrade, amount){
222
      player.global_upgrades_current[upgrade] += amount;
223
      // We cap it between 1 and the current max level
224
      player.global_upgrades_current[upgrade] = Math.max(1, Math.min(player.global_upgrades_current[upgrade], player.global_upgrades[upgrade]));
225
    };
226
227
    ct.visibleUpgrades = function() {
228
      return visibility.visible(data.global_upgrades, upgradeService.filterByTag('redoxes'));
229
    };
230
231
    state.registerUpdate('redox', ct.update);
232
  }
233
]);
234